// Copyright (c) 2020, Compiler Explorer Authors
// All rights reserved.
//
// Redistribution and use in source and binary forms, with or without
// modification, are permitted provided that the following conditions are met:
//
//     * Redistributions of source code must retain the above copyright notice,
//       this list of conditions and the following disclaimer.
//     * Redistributions in binary form must reproduce the above copyright
//       notice, this list of conditions and the following disclaimer in the
//       documentation and/or other materials provided with the distribution.
//
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
// AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
// IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
// ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
// LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
// CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
// SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
// INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
// CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
// ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
// POSSIBILITY OF SUCH DAMAGE.

import express from 'express';

import {isString} from '../../shared/common-utils.js';
import {LanguageKey} from '../../types/languages.interfaces.js';
import {isMobileViewer} from '../app/url-handlers.js';
import {assert} from '../assert.js';
import {ClientState} from '../clientstate.js';
import {ClientStateNormalizer} from '../clientstate-normalizer.js';
import {logger} from '../logger.js';
import {ClientOptionsHandler} from '../options-handler.js';
import {StorageBase} from '../storage/index.js';
import {RenderConfig} from './handler.interfaces.js';
import {cached, csp} from './middleware.js';

export class NoScriptHandler {
    constructor(
        private readonly router: express.Router,
        private readonly clientOptionsHandler: ClientOptionsHandler,
        private readonly renderConfig: RenderConfig,
        private readonly storageHandler: StorageBase,
        private readonly defaultLanguage: string | undefined,
    ) {}

    initializeRoutes() {
        this.router
            .get('/noscript', cached, csp, (req, res) => {
                this.renderNoScriptLayout(undefined, req, res);
            })
            .get('/noscript/z/:id', cached, csp, this.storedStateHandlerNoScript.bind(this))
            .get('/noscript/sponsors', cached, csp, (req, res) => {
                res.render(
                    'noscript/sponsors',
                    this.renderConfig(
                        {
                            embedded: false,
                            mobileViewer: isMobileViewer(req),
                        },
                        req.query,
                    ),
                );
            })
            .get('/noscript/share', cached, csp, this.handleShareLink.bind(this))
            .post('/noscript/share', express.urlencoded({extended: true}), cached, csp, this.handleShareLink.bind(this))
            .get('/noscript/:language', cached, csp, (req, res) => {
                this.renderNoScriptLayout(undefined, req, res);
            });
    }

    storedStateHandlerNoScript(req: express.Request, res: express.Response, next: express.NextFunction) {
        const id = req.params.id;
        this.storageHandler
            .expandId(id)
            .then(result => {
                const config = JSON.parse(result.config);

                let clientstate: ClientState;
                if (config.content) {
                    const normalizer = new ClientStateNormalizer();
                    normalizer.fromGoldenLayout(config);

                    clientstate = normalizer.normalized;
                } else {
                    clientstate = new ClientState(config);
                }

                this.renderNoScriptLayout(clientstate, req, res);

                this.storageHandler.incrementViewCount(id).catch(err => {
                    logger.error(`Error incrementing view counts for ${id} - ${err}`);
                });
            })
            .catch(err => {
                logger.warn(`Could not expand ${id}: ${err}`);
                next({
                    statusCode: 404,
                    message: `ID "${id}" could not be found`,
                });
            });
    }

    createDefaultState(wantedLanguage: LanguageKey) {
        const options = this.clientOptionsHandler.get();

        const state = new ClientState();
        const session = state.findOrCreateSession(1);
        session.language = wantedLanguage;
        if (options.languages[wantedLanguage]) {
            session.source = options.languages[wantedLanguage].example;
        } else {
            session.source = '';
        }

        const compiler = session.findOrCreateCompiler(1);
        if (options.defaultCompiler[wantedLanguage]) {
            compiler.id = options.defaultCompiler[wantedLanguage];
        } else {
            compiler.id = '';
        }

        return state;
    }

    renderNoScriptLayout(state: ClientState | undefined, req: express.Request, res: express.Response) {
        let wantedLanguage = 'c++';
        if (req.params?.language) {
            wantedLanguage = req.params.language;
        } else {
            if (this.defaultLanguage) wantedLanguage = this.defaultLanguage;
            if (req.query.language) {
                const lang = req.query.language;
                assert(isString(lang));
                wantedLanguage = lang;
            }
        }

        if (!state) {
            state = this.createDefaultState(wantedLanguage as LanguageKey);
        }

        res.render(
            'noscript/index',
            this.renderConfig(
                {
                    embedded: false,
                    mobileViewer: isMobileViewer(req),
                    wantedLanguage: wantedLanguage,
                    clientstate: state,
                    storedStateId: req.params.id || false,
                },
                req.query,
            ),
        );
    }

    async handleShareLink(req: express.Request, res: express.Response) {
        // Getting form data with proper type checking - handle both GET and POST
        const source =
            typeof req.body?.source === 'string'
                ? req.body.source
                : typeof req.query.source === 'string'
                  ? req.query.source
                  : '';
        const compiler =
            typeof req.body?.compiler === 'string'
                ? req.body.compiler
                : typeof req.query.compiler === 'string'
                  ? req.query.compiler
                  : '';
        const userArguments =
            typeof req.body?.userArguments === 'string'
                ? req.body.userArguments
                : typeof req.query.userArguments === 'string'
                  ? req.query.userArguments
                  : '';
        const language =
            typeof req.body?.lang === 'string'
                ? req.body.lang
                : typeof req.query.language === 'string'
                  ? req.query.language
                  : 'c++';

        logger.debug('Received data for sharing:', {source, compiler, userArguments, language});

        // Creating a simple state for sharing
        const state = this.createDefaultState(language as LanguageKey);

        if (source) {
            const session = state.findOrCreateSession(1);
            session.source = source;
            session.language = language;

            if (compiler) {
                const compilerObj = session.findOrCreateCompiler(1);
                compilerObj.id = compiler;
            }

            if (userArguments) {
                const compilerObj = session.findOrCreateCompiler(1);
                compilerObj.options = userArguments;
            }
        }

        // Generating shareable URL
        const shareableUrl = await this.generateShareableUrl(state);

        const httpRoot = (this.renderConfig as any).httpRoot || '/';
        const relativeUrl = shareableUrl.substring(shareableUrl.lastIndexOf('/z/') + 1);
        const shortlink = `${req.protocol}://${req.get('host')}${httpRoot}${relativeUrl}`;

        logger.debug('Shareable URL:', shortlink);

        // Rendering the share template
        const renderConfig = this.renderConfig(
            {
                embedded: false,
                mobileViewer: isMobileViewer(req),
                wantedLanguage: language,
                clientstate: state,
                shareableUrl: shortlink,
                source: source,
            },
            req.query,
        );

        // Adding httpRoot to the render config
        (renderConfig as any).httpRoot = httpRoot;

        res.render('noscript/share', renderConfig);
    }

    async generateShareableUrl(state: ClientState): Promise<string> {
        try {
            // Creating the stored object like the main handler does
            const {config, configHash} = StorageBase.getSafeHash(state);

            // Finding or create the unique subhash
            const result = await this.storageHandler.findUniqueSubhash(configHash);

            if (!result.alreadyPresent) {
                const storedObject = {
                    prefix: result.prefix,
                    uniqueSubHash: result.uniqueSubHash,
                    fullHash: configHash,
                    config: config,
                };

                await this.storageHandler.storeItem(storedObject, {} as express.Request);
            }

            return `/z/${result.uniqueSubHash}`;
        } catch (err) {
            logger.error(`Error storing share state: ${err}`);
            // Fallback to direct encoding
            const stateString = JSON.stringify(state);
            const base64State = Buffer.from(stateString).toString('base64url');
            return `/#${base64State}`;
        }
    }
}
